查看原文
其他

Spring 的 Controller 是单例还是多例?怎么保证并发的安全

大家好,我是磊哥。


答案:

controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。正因为单例所以不是线程安全的

我们下面来简单的验证下:

  1. package com.riemann.springbootdemo.controller;

  2. import org.springframework.context.annotation.Scope;

  3. import org.springframework.stereotype.Controller;

  4. import org.springframework.web.bind.annotation.RequestMapping;

  5. /**

  6. * @author riemann

  7. * @date 2019/07/29 22:56

  8. */

  9. @Controller

  10. publicclassScopeTestController{

  11.    privateint num = 0;

  12.    @RequestMapping("/testScope")

  13.    publicvoid testScope() {

  14.        System.out.println(++num);

  15.    }

  16.    @RequestMapping("/testScope2")

  17.    publicvoid testScope2() {

  18.        System.out.println(++num);

  19.    }

  20. }

注 意
 文末有:7701页互联网大厂面试题 

我们首先访问 

http://localhost:8080/testScope,得到的答案是 1

然后我们再访问 

http://localhost:8080/testScope2,得到的答案是 2

得到的不同的值,这是线程不安全的

接下来我们再来给 controller增加作用多例 @Scope("prototype")

  1. package com.riemann.springbootdemo.controller;

  2. import org.springframework.context.annotation.Scope;

  3. import org.springframework.stereotype.Controller;

  4. import org.springframework.web.bind.annotation.RequestMapping;

  5. /**

  6. * @author riemann

  7. * @date 2019/07/29 22:56

  8. */

  9. @Controller

  10. @Scope("prototype")

  11. publicclassScopeTestController{

  12.    privateint num = 0;

  13.    @RequestMapping("/testScope")

  14.    publicvoid testScope() {

  15.        System.out.println(++num);

  16.    }

  17.    @RequestMapping("/testScope2")

  18.    publicvoid testScope2() {

  19.        System.out.println(++num);

  20.    }

  21. }

我们依旧首先访问 

http://localhost:8080/testScope,得到的答案是 1

然后我们再访问 

http://localhost:8080/testScope2,得到的答案还是 1

单例是不安全的,会导致属性重复使用

解决方案

1、不要在controller中定义成员变量
2、万一必须要定义一个非静态成员变量时候,则通过注解 @Scope(“prototype”),将其设置为多例模式
3、在Controller中使用ThreadLocal变量

补充说明

spring bean作用域有以下5个:

singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理;

prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;

下面是在web项目下才用到的

request:搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;

session:每次会话,同上;

global session:全局的web域,类似于servlet中的application。

近期技术热文


往期推荐



说实话,DataGrip真得牛逼,只是你不会用而已~

看看人家,那后端API接口写得,那叫一个优雅!

卧槽:这款 IDEA 插件能搞,流程图、架构图,N种图... 简直神器!

免费版 IDEA 为啥不能使用 Tomcat ?


第3版:互联网大厂面试题

包括 Java 集合、JVM、多线程、并发编程、设计模式、算法调优、Spring全家桶、Java、MyBatis、ZooKeeper、Dubbo、Elasticsearch、Memcached、MongoDB、Redis、MySQL、RabbitMQ、Kafka、Linux、Netty、Tomcat、Python、HTML、CSS、Vue、React、JavaScript、Android 大数据、阿里巴巴等大厂面试题等、等技术栈!

阅读原文: 高清 7701页大厂面试题  PDF

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存